---
-- @class PANEL
-- @desc Scoreboard player score row, based on sandbox version
-- @section TTTScorePlayerRow
---

ttt_include("vgui__cl_sb_info")

local TryT = LANG.TryTranslation
local GetTranslation = LANG.GetTranslation
local math = math
local table = table
local IsValid = IsValid
local surface = surface
local vgui = vgui
local drawRoundedBox = draw.RoundedBox

local colorTransparent = Color(35, 39, 46, 0)        -- 现代深色透明背景

local materialIndicatorDev = "vgui/ttt/ttt2_indicator_dev"
local materialIndicatorVIP = "vgui/ttt/ttt2_indicator_vip"
local materialIndicatorAddonDev = "vgui/ttt/ttt2_indicator_addondev"
local materialIndicatorAdmin = "vgui/ttt/ttt2_indicator_admin"
local materialIndicatorStreamer = "vgui/ttt/ttt2_indicator_streamer"
local materialIndicatorHeroes = "vgui/ttt/ttt2_indicator_heroes"

local materialNoTeam = "vgui/ttt/dynamic/roles/icon_no_team"

local dev_tbl = {
    ["76561197964193008"] = true, -- Bad King Urgrain
    ["76561198049831089"] = true, -- Alf21
    ["76561198058039701"] = true, -- saibotk
    ["76561198047819379"] = true, -- Mineotopia
    ["76561198052323988"] = true, -- LeBroomer
    ["76561198076404571"] = true, -- Histalek
    ["76561198296468397"] = true, -- ZenBre4ker
}

local vip_tbl = {
    ["76561198378608300"] = true, -- SirKadeeka / $rã£
    ["76561198102780049"] = true, -- Nick
    ["76561198020955880"] = true, -- V3kta
    ["76561198033468770"] = true, -- dok441
    ["76561198208729341"] = true, -- eBBIS
    ["76561198152558244"] = true, -- Lost TheSuspect
    ["76561198162764168"] = true, -- DerJohn
    ["76561198132229662"] = true, -- Satton(RU)
    ["76561198007725535"] = true, -- Skatcat
    ["76561197989909602"] = true, -- Tobiti
    ["76561198150260014"] = true, -- Lunex
    ["76561198076404571"] = true, -- Histalek
    ["76561198042086461"] = true, -- James
    ["76561193814529882"] = true, -- Trystan
    ["76561198114719750"] = true, -- Reispfannenfresser
    ["76561198082931319"] = true, -- Henk
    ["76561198049910438"] = true, -- Zzzaaaccc13
    ["76561198296468397"] = true, -- ZenBre4ker
    ["76561198041170748"] = true, -- Pytho | Paul
    ["76561197999466165"] = true, -- EntranceJew
    ["76561198279816989"] = true, -- mexikoedi
}

local addondev_tbl = {
    ["76561198102780049"] = true, -- Nick
    ["76561197989909602"] = true, -- Tobiti
    ["76561198076404571"] = true, -- Histalek
    ["76561198027025876"] = true, -- 8Z
}

---
-- Add an add-on developer
-- @param string steamid64
-- @realm client
function AddTTT2AddonDev(steamid64)
    if not steamid64 then
        return
    end

    addondev_tbl[tostring(steamid64)] = true
end

local streamer_tbl = {
    ["76561198049831089"] = true,
    ["76561198058039701"] = true,
    ["76561198047819379"] = true, -- Mineotopia
    ["76561198052323988"] = true,
}

--TODO: move into heroes
local heroes_tbl = {
    ["76561198000950884"] = true, -- Dhalucard
}

local namecolor = {
    default = Color(240, 245, 250, 255),      -- 现代浅色文字
    dev = Color(100, 240, 105, 255),
    vip = Color(220, 55, 55, 255),
    addondev = Color(30, 105, 30, 255),
    admin = Color(255, 210, 35, 255),
    streamer = Color(100, 70, 140, 255),
    heroes = Color(70, 125, 110, 255),
}

SB_ROW_HEIGHT = 24 --16

local iconSizes = 16

local PANEL = {}

local function _func1(ply)
    return ply:Ping()
end

local function _func2(ply)
    return ply:Deaths()
end

local function _func3(ply)
    return ply:Frags()
end

local function _func4(ply)
    return math.Round(ply:GetBaseKarma())
end

---
-- @ignore
function PANEL:Init()
    -- cannot create info card until player state is known
    self.info = nil
    self.open = false
    self.cols = {}

    self:AddColumn(GetTranslation("sb_ping"), _func1)
    self:AddColumn(GetTranslation("sb_deaths"), _func2)
    self:AddColumn(GetTranslation("sb_score"), _func3)

    if KARMA.IsEnabled() then
        self:AddColumn(GetTranslation("sb_karma"), _func4)
    end

    ---
    -- Let hooks add their custom columns
    -- @realm client
    hook.Run("TTTScoreboardColumns", self)

    for i = 1, #self.cols do
        self.cols[i]:SetMouseInputEnabled(false)
    end

    self.tag = vgui.Create("DLabel", self)
    self.tag:SetText("")
    self.tag:SetMouseInputEnabled(false)

    self.sresult = vgui.Create("DImage", self)
    self.sresult:SetSize(iconSizes, iconSizes)
    self.sresult:SetMouseInputEnabled(true)

    self.dev = vgui.Create("DImage", self)
    self.dev:SetSize(iconSizes, iconSizes)
    self.dev:SetMouseInputEnabled(true)
    self.dev:SetKeepAspect(true)
    self.dev:SetTooltip(TryT("sb_rank_tooltip_developer"))

    self.vip = vgui.Create("DImage", self)
    self.vip:SetSize(iconSizes, iconSizes)
    self.vip:SetMouseInputEnabled(true)
    self.vip:SetKeepAspect(true)
    self.vip:SetTooltip(TryT("sb_rank_tooltip_vip"))

    self.addondev = vgui.Create("DImage", self)
    self.addondev:SetSize(iconSizes, iconSizes)
    self.addondev:SetMouseInputEnabled(true)
    self.addondev:SetKeepAspect(true)
    self.addondev:SetTooltip(TryT("sb_rank_tooltip_addondev"))

    self.admin = vgui.Create("DImage", self)
    self.admin:SetSize(iconSizes, iconSizes)
    self.admin:SetMouseInputEnabled(true)
    self.admin:SetKeepAspect(true)
    self.admin:SetTooltip(TryT("sb_rank_tooltip_admin"))

    self.streamer = vgui.Create("DImage", self)
    self.streamer:SetSize(iconSizes, iconSizes)
    self.streamer:SetMouseInputEnabled(true)
    self.streamer:SetKeepAspect(true)
    self.streamer:SetTooltip(TryT("sb_rank_tooltip_streamer"))

    self.heroes = vgui.Create("DImage", self)
    self.heroes:SetSize(iconSizes, iconSizes)
    self.heroes:SetMouseInputEnabled(true)
    self.heroes:SetKeepAspect(true)
    self.heroes:SetTooltip(TryT("sb_rank_tooltip_heroes"))

    self.avatar = vgui.Create("AvatarImage", self)
    self.avatar:SetSize(SB_ROW_HEIGHT, SB_ROW_HEIGHT)
    self.avatar:SetMouseInputEnabled(false)

    self.nick2 = vgui.Create("DLabel", self)
    self.nick2:SetMouseInputEnabled(false)

    self.nick3 = vgui.Create("DLabel", self)
    self.nick3:SetMouseInputEnabled(false)

    self.nick = vgui.Create("DLabel", self)
    self.nick:SetMouseInputEnabled(false)

    self.team2 = vgui.Create("DImage", self)
    self.team2:SetSize(iconSizes, iconSizes)
    self.team2:SetMouseInputEnabled(false)
    self.team2:SetKeepAspect(true)

    self.team = vgui.Create("DImage", self)
    self.team:SetSize(iconSizes, iconSizes)
    self.team:SetMouseInputEnabled(true)
    self.team:SetKeepAspect(true)
    self.team:SetTooltip(TryT("sb_rank_tooltip_team"))

    self.voice = vgui.Create("DImageButton", self)
    self.voice:SetSize(iconSizes, iconSizes)

    self:SetCursor("hand")
end

---
-- Adds a column
-- @param string label
-- @param function func
-- @param number width
-- @return Panel DLabel
-- @realm client
function PANEL:AddColumn(label, func, width)
    local lbl = vgui.Create("DLabel", self)
    lbl.GetPlayerText = func
    lbl.IsHeading = false
    lbl.Width = width or 50 -- Retain compatibility with existing code

    self.cols[#self.cols + 1] = lbl

    return lbl
end

---
-- Mirror sb_main, of which it and this file both call using the
-- TTTScoreboardColumns hook, but it is useless in this file.
-- Exists only so the hook wont return an error if it tries to
-- use the AddFakeColumn function of `sb_main`, which would
-- cause this file to raise a `function not found` error or others
-- @realm client
function PANEL:AddFakeColumn() end

local function ColorForPlayer(ply)
    if IsValid(ply) then
        ---
        -- @realm client
        local c = hook.Run("TTTScoreboardColorForPlayer", ply)

        -- verify that we got a proper color
        if c and istable(c) and c.r and c.b and c.g and c.a then
            return c
        else
            ErrorNoHaltWithStack(
                "TTTScoreboardColorForPlayer hook returned something that isn't a color!\n"
            )
        end
    end

    return namecolor.default
end

---
-- @param number width
-- @param number height
-- @realm client
-- @ignore
function PANEL:Paint(width, height)
    if not IsValid(self.Player) then
        return false
    end

    local ply = self.Player

    ---
    -- @realm client
    local c = hook.Run("TTTScoreboardRowColorForPlayer", ply)

    -- 现代化行背景
    if c and c.a > 0 then
        draw.RoundedBox(6, 0, 0, width, SB_ROW_HEIGHT, c)
    end

    -- 当前玩家高亮效果 - 现代化设计
    if ply == LocalPlayer() then
        local pulse = math.abs(math.sin(RealTime() * 1.5)) * 0.3 + 0.1
        local highlightColor = Color(52, 152, 219, math.Clamp(pulse * 255, 50, 150))
        draw.RoundedBox(6, 1, 1, width - 2, SB_ROW_HEIGHT - 2, highlightColor)
        
        -- 边框效果
        surface.SetDrawColor(52, 152, 219, 200)
        surface.DrawOutlinedRect(0, 0, width, SB_ROW_HEIGHT)
    end

    return true
end

---
-- @param Player ply
-- @realm client
function PANEL:SetPlayer(ply)
    self.Player = ply
    self.avatar:SetPlayer(ply)

    local client = LocalPlayer()

    if ply ~= client then
        self.voice:SetTooltip(GetTranslation("scoreboard_voice_tooltip"))
    end

    if not self.info then
        local g = ScoreGroup(ply)

        if g == GROUP_TERROR and ply ~= LocalPlayer() then
            self.info = vgui.Create("TTTScorePlayerInfoTags", self)
            self.info:SetPlayer(ply)

            self:InvalidateLayout()
        else
            self.info = vgui.Create("TTTScorePlayerInfoSearch", self)
            self.info:SetPlayer(ply)

            self:InvalidateLayout()
        end
    else
        self.info:SetPlayer(ply)

        self:InvalidateLayout()
    end

    self.voice.DoClick = function()
        if IsValid(ply) and ply ~= client then
            local muted = VOICE.GetPreferredPlayerVoiceMuted(ply)
            VOICE.SetPreferredPlayerVoiceMuted(ply, not muted)
            VOICE.UpdatePlayerVoiceVolume(ply)
        end
    end

    self.voice.OnMouseWheeled = function(label, delta)
        if IsValid(ply) and ply ~= client then
            self:ScrollPlayerVolume(delta)
        end
    end

    self:UpdatePlayerData()
end

---
-- @return Player
-- @realm client
function PANEL:GetPlayer()
    return self.Player
end

---
-- @realm client
function PANEL:UpdatePlayerData()
    if not IsValid(self.Player) then
        return
    end

    local ply = self.Player

    for i = 1, #self.cols do
        -- Set text from function, passing the label along so stuff like text
        -- color can be changed
        self.cols[i]:SetText(self.cols[i].GetPlayerText(ply, self.cols[i]))
    end

    self.nick2:SetText(ply:Nick())
    self.nick2:SizeToContents()
    self.nick2:SetTextColor(Color(45, 50, 60, 255))       -- 现代深色阴影

    self.nick3:SetText(ply:Nick())
    self.nick3:SizeToContents()
    self.nick3:SetTextColor(Color(45, 50, 60, 255))       -- 现代深色阴影

    self.nick:SetText(ply:Nick())
    self.nick:SizeToContents()
    self.nick:SetTextColor(ColorForPlayer(ply))

    local tm = ply:GetTeam() or nil
    if tm then
        local tmData = TEAMS[tm]
        if tm == TEAM_NONE or not tmData or tmData.alone then
            self.team2:SetImage(materialNoTeam)
            self.team:SetImage(materialNoTeam)
        else
            local teamImageName = tmData.iconMaterial:GetName()

            self.team2:SetImage(teamImageName)
            self.team:SetImage(teamImageName)
        end

        self.team2:SetImageColor(Color(45, 50, 60, 255))     -- 现代深色
        self.team:SetImageColor(Color(240, 245, 250, 255))   -- 现代浅色

        self.team:SetTooltip(LANG.GetTranslation(tm))
    end

    local showTeam = ply:HasRole()

    self.team2:SetVisible(showTeam)
    self.team:SetVisible(showTeam)

    local steamid64 = ply:SteamID64()
    if steamid64 then
        steamid64 = tostring(steamid64)
    end

    local isdev = steamid64 and dev_tbl[steamid64] == true

    self.dev:SetVisible(isdev and GetGlobalBool("ttt_highlight_dev", true))

    if not isdev or not GetGlobalBool("ttt_highlight_dev", true) then
        self.vip:SetVisible(
            steamid64 and vip_tbl[steamid64] and GetGlobalBool("ttt_highlight_vip", true)
        )
        self.addondev:SetVisible(
            steamid64 and addondev_tbl[steamid64] and GetGlobalBool("ttt_highlight_addondev", true)
        )
    else
        self.vip:SetVisible(false)
        self.addondev:SetVisible(false)
    end

    self.admin:SetVisible(admin.IsAdmin(ply) and GetGlobalBool("ttt_highlight_admins", true))
    self.streamer:SetVisible(
        steamid64 and streamer_tbl[steamid64] and GetGlobalBool("ttt_highlight_supporter", true)
    )
    self.heroes:SetVisible(
        steamid64 and heroes_tbl[steamid64] and GetGlobalBool("ttt_highlight_supporter", true)
    )

    local ptag = ply.sb_tag

    if ScoreGroup(ply) ~= GROUP_TERROR then
        ptag = nil
    end

    self.tag:SetText(ptag and GetTranslation(ptag.txt) or "")
    self.tag:SetTextColor(ptag and ptag.color or Color(180, 185, 195, 255))  -- 现代次要文字色
    self.tag:SetFont("treb_small")

    local show_sresult = true
    -- we have body search results
    if ply.bodySearchResult and ply.bodySearchResult.base.isPublicPolicingSearch then
        self.sresult:SetImage("icon16/magnifier.png")
        self.sresult:SetTooltip(GetTranslation("cl_sb_row_sresult_pub_police"))
    elseif ply.bodySearchResult and ply.bodySearchResult.show_sb then
        self.sresult:SetImage("icon16/information.png")
        self.sresult:SetTooltip(GetTranslation("cl_sb_row_sresult_direct_conf"))
    else
        show_sresult = false
    end

    self.sresult:SetVisible(show_sresult)

    -- cols are likely to need re-centering
    self:LayoutColumns()

    if self.info then
        self.info:UpdatePlayerData()
    end

    if self.Player ~= LocalPlayer() then
        local muted = VOICE.GetPreferredPlayerVoiceMuted(self.Player)

        self.voice:SetImage(muted and "icon16/sound_mute.png" or "icon16/sound.png")
    else
        self.voice:Hide()
    end
end

---
-- @ignore
function PANEL:ApplySchemeSettings()
    local ply = self.Player

    for i = 1, #self.cols do
        local v = self.cols[i]

        v:SetFont("treb_small")
        v:SetTextColor(Color(180, 185, 195, 255))         -- 现代次要文字色
    end

    self.nick2:SetFont("treb_small")
    self.nick2:SetTextColor(Color(45, 50, 60, 255))       -- 现代深色阴影

    self.nick3:SetFont("treb_small")
    self.nick3:SetTextColor(Color(45, 50, 60, 255))       -- 现代深色阴影

    self.nick:SetFont("treb_small")

    if IsValid(ply) then
        self.nick:SetTextColor(ColorForPlayer(ply))
    end

    local ptag = IsValid(ply) and ply.sb_tag or nil
    self.tag:SetTextColor(ptag and ptag.color or Color(180, 185, 195, 255))  -- 现代次要文字色
    self.tag:SetFont("treb_small")

    self.sresult:SetImageColor(Color(180, 185, 195, 255)) -- 现代次要图标色

    self.dev:SetImage(materialIndicatorDev)
    self.dev:SetImageColor(namecolor.dev)

    self.vip:SetImage(materialIndicatorVIP)
    self.vip:SetImageColor(namecolor.vip)

    self.addondev:SetImage(materialIndicatorAddonDev)
    self.addondev:SetImageColor(namecolor.addondev)

    self.admin:SetImage(materialIndicatorAdmin)
    self.admin:SetImageColor(namecolor.admin)

    self.streamer:SetImage(materialIndicatorStreamer)
    self.streamer:SetImageColor(namecolor.streamer)

    self.heroes:SetImage(materialIndicatorHeroes)
    self.heroes:SetImageColor(namecolor.heroes)
end

---
-- @realm client
function PANEL:LayoutColumns()
    local cx = self:GetWide()

    for i = 1, #self.cols do
        local v = self.cols[i]

        v:SizeToContents()

        cx = cx - v.Width

        v:SetPos(cx - v:GetWide() * 0.5, (SB_ROW_HEIGHT - v:GetTall()) * 0.5)
    end

    self.tag:SizeToContents()

    cx = cx - 90

    self.tag:SetPos(cx - self.tag:GetWide() * 0.5, (SB_ROW_HEIGHT - self.tag:GetTall()) * 0.5)

    self.sresult:SetPos(cx - 8, (SB_ROW_HEIGHT - iconSizes) * 0.5)

    local x = self.nick:GetPos()
    local w = self.nick:GetSize()

    local count = 0
    local mgn = 10

    local tx = x + w + mgn
    local ty = (SB_ROW_HEIGHT - iconSizes) * 0.5

    local iconTbl = {
        "dev",
        "vip",
        "addondev",
        "admin",
        "streamer",
        "heroes",
    }

    for i = 1, #iconTbl do
        local entry = iconTbl[i]
        local iconData = self[entry]

        if iconData:IsVisible() then
            iconData:SetPos(tx + (iconSizes + mgn) * count, ty)

            count = count + 1
        end
    end
end

---
-- @ignore
function PANEL:PerformLayout()
    self.avatar:SetPos(0, 0)
    self.avatar:SetSize(SB_ROW_HEIGHT, SB_ROW_HEIGHT)

    local fw = sboard_panel.ply_frame:GetWide()

    self:SetWide(sboard_panel.ply_frame.scroll.Enabled and (fw - iconSizes) or fw)

    if not self.open then
        self:SetSize(self:GetWide(), SB_ROW_HEIGHT)

        if self.info then
            self.info:SetVisible(false)
        end
    elseif self.info then
        self:SetSize(self:GetWide(), 100 + SB_ROW_HEIGHT)

        self.info:SetVisible(true)
        self.info:SetPos(5, SB_ROW_HEIGHT + 5)
        self.info:SetSize(self:GetWide(), 100)
        self.info:PerformLayout()

        self:SetSize(self:GetWide(), SB_ROW_HEIGHT + self.info:GetTall())
    end

    local tx = SB_ROW_HEIGHT + 10 + iconSizes
    local ty = (SB_ROW_HEIGHT - self.nick:GetTall()) * 0.5

    self.nick2:SizeToContents()
    self.nick2:SetPos(tx + 1, ty + 1)

    self.nick3:SizeToContents()
    self.nick3:SetPos(tx, ty)

    self.nick:SizeToContents()
    self.nick:SetPos(tx, ty)

    self.team2:SetPos(tx - iconSizes - 4, (SB_ROW_HEIGHT - iconSizes) * 0.5 + 1)
    self.team:SetPos(tx - iconSizes - 5, (SB_ROW_HEIGHT - iconSizes) * 0.5)

    self:LayoutColumns()

    self.voice:SetVisible(not self.open)
    self.voice:SetSize(iconSizes, iconSizes)
    self.voice:DockMargin(4, 4, 4, 4)
    self.voice:Dock(RIGHT)
end

---
-- @param number x
-- @param number y
-- @realm client
function PANEL:DoClick(x, y)
    self:SetOpen(not self.open)
end

---
-- @param boolean o
-- @realm client
function PANEL:SetOpen(o)
    if self.open then
        surface.PlaySound("ui/buttonclickrelease.wav")
    else
        surface.PlaySound("ui/buttonclick.wav")
    end

    self.open = o

    self:PerformLayout()
    self:GetParent():PerformLayout()

    sboard_panel:PerformLayout()
end

---
-- @realm client
function PANEL:DoRightClick()
    local menu = DermaMenu()
    menu.Player = self:GetPlayer()

    ---
    -- @realm client
    local close = hook.Run("TTTScoreboardMenu", menu)
    if close then
        menu:Remove()

        return
    end

    menu:Open()
end

---
-- @param number delta
-- @realm client
function PANEL:ScrollPlayerVolume(delta)
    local ply = self:GetPlayer()

    -- Bots return nil for the steamid64 on the client, so we need to improvise a bit
    local identifier = ply:IsBot() and ply:Nick() or ply:SteamID64()

    local cur_volume = VOICE.GetPreferredPlayerVoiceVolume(ply)
    cur_volume = cur_volume ~= nil and cur_volume or 1

    local new_volume = delta == -1 and math.max(0, cur_volume - 0.01)
        or math.min(1, cur_volume + 0.01)
    new_volume = math.Round(new_volume, 2)

    VOICE.SetPreferredPlayerVoiceVolume(ply, new_volume)
    VOICE.UpdatePlayerVoiceVolume(ply)

    if self.voice.percentage_frame ~= nil and not self.voice.percentage_frame:IsVisible() then
        self.voice.percentage_frame:Show()
    end

    local x, y = input.GetCursorPos()
    local width = 60
    local height = 40
    local padding = 10

    -- Frame
    local frame = self.voice.percentage_frame ~= nil and self.voice.percentage_frame
        or vgui.Create("DFrame")
    frame:SetPos(x - 10, y + 25)
    frame:SetSize(width, height)
    frame:SetTitle("")
    frame:ShowCloseButton(false)
    frame:SetDraggable(false)
    frame:SetSizable(false)
    frame.Paint = function(_, w, h)
        drawRoundedBox(8, 0, 0, w, h, Color(24, 25, 28, 180))
    end

    self.voice.percentage_frame = frame

    local label = self.voice.percentage_frame_label ~= nil and self.voice.percentage_frame_label
        or vgui.Create("DLabel", frame)
    label:SetPos(padding, padding)
    label:SetFont("treb_small")
    label:SetSize(width - padding * 2, 20)
    label:SetColor(COLOR_WHITE)
    label:SetText(tostring(math.Round(new_volume * 100)) .. "%")

    self.voice.percentage_frame_label = label

    timer.Remove("ttt_score_close_perc_frame_" .. identifier)

    timer.Create("ttt_score_close_perc_frame_" .. identifier, 1.5, 1, function()
        if not self.voice and frame and frame:IsVisible() then
            frame:Close()
            frame = nil

            return
        end

        if
            not self.voice
            or not self.voice.percentage_frame
            or not self.voice.percentage_frame:IsVisible()
        then
            return
        end

        self.voice.percentage_frame:Hide()
    end)

    hook.Add("ScoreboardHide", "TTTCloseVolumeFrame_" .. identifier, function()
        if not self.voice and frame and frame:IsVisible() then
            frame:Close()
            frame = nil

            return
        end

        if
            not self.voice
            or not self.voice.percentage_frame
            or not self.voice.percentage_frame:IsVisible()
        then
            return
        end

        self.voice.percentage_frame:Hide()

        timer.Remove("ttt_score_close_perc_frame_" .. identifier)
    end)
end

vgui.Register("TTTScorePlayerRow", PANEL, "DButton")

---
-- Updates the row background color for a @{Player} in the scoreboard.
-- @note TTT2 also uses this hook for the default colours, so you should return nil for players
-- that you do not want to do anything special with.
-- @param Player ply The player whose color should be changed
-- @return nil|Color The color highlight for the row background in the scoreboard
-- @hook
-- @realm client
function GM:TTTScoreboardRowColorForPlayer(ply)
    local col = colorTransparent

    if IsValid(ply) and ply.HasRole and ply:HasRole() then
        col = table.Copy(ply:GetRoleColor())
        col.a = 255 -- old value: 30
    end

    return col
end

---
-- Called to determine what colour a player's name should be on the scoreboard.
-- @note TTT2 also uses this hook for the default colours, so you should return nil for players
-- that you do not want to do anything special with.
-- @param Player ply The player whose color should be changed
-- @return nil|Color The color highlight for the name in the scoreboard
-- @hook
-- @realm client
function GM:TTTScoreboardColorForPlayer(ply)
    if IsValid(ply) then
        local steamid64 = ply:SteamID64()

        if steamid64 then
            steamid64 = tostring(steamid64)

            if dev_tbl[steamid64] and GetGlobalBool("ttt_highlight_dev", true) then
                return namecolor.dev
            elseif vip_tbl[steamid64] and GetGlobalBool("ttt_highlight_vip", true) then
                return namecolor.vip
            elseif addondev_tbl[steamid64] and GetGlobalBool("ttt_highlight_addondev", true) then
                return namecolor.addondev
            end
        end

        if admin.IsAdmin(ply) and GetGlobalBool("ttt_highlight_admins", true) then
            return namecolor.admin
        end
    end

    return namecolor.default
end

---
-- Called to show a context menu when the player right-clicks a player in the scoreboard.
-- @note By default there is no context menu when right clicking, it will only appear
-- if you add options in this hook.
-- @param DermaMenu menu A @{DermaMenu} that you can add options to that the player can click
-- @hook
-- @realm client
function GM:TTTScoreboardMenu(menu) end
